<script setup lang="ts">
import api from '@/api';
import { Activity } from '@/types/activity';
import { unexpectedError } from '@/utils/unexpected-error';
import type { User } from '@directus/types';
import { ref, watch } from 'vue';
import CommentInput from './comment-input.vue';
import CommentItemHeader from './comment-item-header.vue';

const props = withDefaults(
	defineProps<{
		activity: Activity & {
			display: string;
			user: Pick<User, 'id' | 'email' | 'first_name' | 'last_name' | 'avatar'>;
		};
		refresh: () => Promise<void>;
		collection: string;
		primaryKey: string | number;
		userPreviews: Record<string, any>;
	}>(),
	{
		userPreviews: () => ({}),
	},
);

const { editing, cancelEditing } = useEdits();

function useEdits() {
	const edits = ref(props.activity.comment);
	const editing = ref(false);
	const savingEdits = ref(false);

	watch(
		() => props.activity,
		() => (edits.value = props.activity.comment),
	);

	return { edits, editing, savingEdits, saveEdits, cancelEditing };

	async function saveEdits() {
		savingEdits.value = true;

		try {
			await api.patch(`/activity/comment/${props.activity.id}`, {
				comment: edits.value,
			});

			props.refresh();
		} catch (error) {
			unexpectedError(error);
		} finally {
			savingEdits.value = false;
			editing.value = false;
		}
	}

	function cancelEditing() {
		edits.value = props.activity.comment;
		editing.value = false;
	}
}
</script>

<template>
	<div class="comment-item">
		<comment-item-header :refresh="refresh" :activity="activity" @edit="editing = true" />

		<comment-input
			v-if="editing"
			:existing-comment="activity"
			:primary-key="primaryKey"
			:collection="collection"
			:refresh="refresh"
			:previews="userPreviews"
			@cancel="cancelEditing"
		/>

		<div v-else v-md="{ value: activity.display, target: '_blank' }" class="content selectable" />
	</div>
</template>

<style lang="scss" scoped>
.comment-item {
	position: relative;
	margin-bottom: 8px;
	padding: 8px;
	background-color: var(--theme--background);
	border-radius: var(--theme--border-radius);
}

.comment-item:last-of-type {
	margin-bottom: 8px;
}

.comment-item .content {
	display: inline-block;
	max-height: 300px;
	overflow-y: auto;
	min-width: 100%;
	max-width: 100%;
	margin-bottom: -6px;
	line-height: 1.4;
}

.comment-item .content :deep(> *:first-child),
.comment-item .content :deep(p > *:first-child) {
	margin-top: 0;
}

.comment-item .content :deep(> *:last-child),
.comment-item .content :deep(p > *:last-child) {
	margin-bottom: 0;
}

.comment-item .content :deep(a) {
	color: var(--theme--primary);
}

.comment-item .content :deep(blockquote) {
	margin: 8px 0;
	padding-left: 6px;
	color: var(--theme--foreground-subdued);
	font-style: italic;
	border-left: 2px solid var(--theme--form--field--input--border-color);
}

.comment-item .content :deep(img) {
	max-width: 100%;
	margin: 8px 0;
	border-radius: var(--theme--border-radius);
	display: block;
}

.comment-item .content :deep(hr) {
	height: 2px;
	margin: 12px 0;
	border: 0;
	border-top: 2px solid var(--theme--form--field--input--border-color);
}

.comment-item .content :deep(mark) {
	display: inline-block;
	padding: 2px 4px;
	color: var(--theme--primary);
	line-height: 1;
	background: var(--theme--primary-background);
	border-radius: var(--theme--border-radius);
	user-select: text;
	pointer-events: none;
}

.comment-item .content :deep(pre) {
	padding: 2px 4px;
	color: var(--theme--foreground);
	background-color: var(--theme--background-normal);
	border-radius: var(--theme--border-radius);
	margin: 2px 0;
	font-family: var(--theme--fonts--monospace--font-family);
	white-space: nowrap;
	max-width: 100%;
	overflow-x: auto;
}

.comment-item .content :deep(code) {
	padding: 2px 4px;
	color: var(--theme--foreground);
	background-color: var(--theme--background-normal);
	border-radius: var(--theme--border-radius);
	margin: 2px 0;
	font-family: var(--theme--fonts--monospace--font-family);
}

.comment-item .content :deep(pre > code) {
	padding: 0;
	margin: 0;
	white-space: pre;
}

.comment-item .content :deep(:is(h1, h2, h3, h4, h5, h6)) {
	margin-top: 12px;
	font-weight: 600;
	font-size: 16px;
	color: var(--theme--foreground-accent);
}

.comment-item.expand .content::after {
	position: absolute;
	right: 0;
	bottom: 4px;
	left: 0;
	z-index: 1;
	height: 40px;
	background: linear-gradient(
		180deg,
		rgb(var(--background-page-rgb), 0) 0%,
		rgb(var(--background-page-rgb), 0.8) 25%,
		rgb(var(--background-page-rgb), 1) 100%
	);
	content: '';
}

.comment-item.expand .content .expand-text {
	position: absolute;
	right: 0;
	bottom: 8px;
	left: 0;
	z-index: 2;
	height: 24px;
	text-align: center;
	cursor: pointer;
}

.comment-item.expand .content .expand-text span {
	padding: 4px 12px 5px;
	color: var(--theme--foreground-subdued);
	font-weight: 600;
	font-size: 12px;
	background-color: var(--theme--background-normal);
	border-radius: 12px;
	transition:
		color var(--fast) var(--transition),
		background-color var(--fast) var(--transition);
}

.comment-item.expand .content .expand-text:hover span {
	color: var(--foreground-inverted);
	background-color: var(--theme--primary);
}

.comment-item:hover :deep(.comment-header .header-right .time) {
	opacity: 0;
}

.comment-item:hover :deep(.comment-header .header-right .more) {
	opacity: 1;
}

.user-name {
	color: var(--theme--primary);
}

.buttons {
	position: absolute;
	right: 8px;
	bottom: 8px;
}

.cancel {
	margin-right: 4px;
}
</style>
